linux kernel 怎么读cpu写寄存器 inw |
您所在的位置:网站首页 › cdev_add count › linux kernel 怎么读cpu写寄存器 inw |
arm裸机下读写寄存器很容易,各个寄存器和内存的地址是单一地址空间,他们是用相同的指令进行读写操作的.而在linux下就要复杂很多,因为linux支持多个体系架构的CPU。比如arm和x86就不一样,具体的差别我暂时也说不上来,这个涉及到CPU体系的设计。目前我只关心:linux为了支持多个硬件体系,在IO访问上做了自己的接口。可以通过IO内存和IO端口这两种方式进行IO访问。在LED的例子上给出这两种方式的具体实现: 1.利用IO Port的方式: [cpp] view plain copy #include linux/module.h #include linux/moduleparam.h #include linux/init.h
#include linux/kernel.h/* printk() */ #include linux/slab.h /* kmalloc() */ #include linux/fs.h/* everything... */ #include linux/errno.h /* error codes */ #include linux/types.h /* size_t */ #include linux/proc_fs.h #include linux/fcntl.h /* O_ACCMODE */ #include linux/seq_file.h #include linux/cdev.h #include linux/ioport.h
#include mach/regs-gpio.h #include asm/system.h /* cli(), *_flags */ #include asm/uaccess.h /* copy_*_user */ #include asm/io.h
#define LED_NUM 4
struct led_dev { struct cdev dev unsigned port unsigned long offset }
struct led_dev led[4] dev_t dev = 0 static struct resource *led_resource int led_open(struct inode *inode, struct file *filp) { struct led_dev *led/* device information */
led = container_of(inode-i_cdev, struct led_dev, dev) filp-private_data = led/* for other methods */
return 0 /* success */ }
int led_release(struct inode *inode, struct file *filp) { return 0 }
ssize_t led_read(struct file *filp, char __user *buf, size_t count, loff_t *f_pos) { return 0 }
ssize_t led_write(struct file *filp, const char __user *buf, size_t count, loff_t *f_pos) { char data struct led_dev *led u32 value printk(KERN_INFO "debug by baikal: led dev write\n")
led = (struct led_dev *)filp-private_data copy_from_user(data,buf,count) if(data == '0') { printk(KERN_INFO "debug by baikal: led off\n") value = inl((unsigned)(S3C2410_GPBDAT)) outl(value | 1led-offset,(unsigned)(S3C2410_GPBDAT)) //value = ioread32(led-base) //iowrite32( value | 1led-offset, led-base) } else { printk(KERN_INFO "debug by baikal: led on\n") value = inl((unsigned)(S3C2410_GPBDAT)) outl(value ~(1led-offset),(unsigned)(S3C2410_GPBDAT)) //value = ioread32(led-base) //iowrite32( value ~(1led-offset), led-base) } }
struct file_operations led_fops = { .owner =THIS_MODULE, .read = led_read, .write =led_write, //.ioctl =led_ioctl, .open = led_open, .release = led_release, }
static int led_init(void) { int result, i result = alloc_chrdev_region(dev, 0, LED_NUM,"LED") if (result 0) { printk(KERN_WARNING "LED: can't get major %d\n", MAJOR(dev)) return result } led_resource = request_region(0x56000014,0x4,"led") if(led_resource == NULL) { printk(KERN_ERR " Unable to register LED I/O addresses\n") return -1 } for(i = 0i LED_NUMi++) { cdev_init( led[i].dev, led_fops) //led[i].port = ioport_map(0x56000014,0x4) //led[i].base = ioremap(0x56000014,0x4) led[i].offset = i + 5 //leds GPB5\6\7\8 led[i].dev.owner = THIS_MODULE led[i].dev.ops = led_fops result = cdev_add(led[i].dev,MKDEV(MAJOR(dev),i),1) if(result 0) { printk(KERN_ERR "LED: can't add led%d\n",i) return result } }
return 0 }
static void led_exit(void) { int i release_region(0x56000014,0x4) for( i = 0i LED_NUMi++) { //iounmap(led[i].base)
cdev_del(led[i].dev) } unregister_chrdev_region(dev, LED_NUM)
} module_init(led_init) module_exit(led_exit)
MODULE_AUTHOR("Baikal") MODULE_LICENSE("GPL") MODULE_DESCRIPTION("Simple LED Driver") 2.利用IO Mem的方式: [cpp] view plain copy #include linux/module.h #include linux/moduleparam.h #include linux/init.h
#include linux/kernel.h/* printk() */ #include linux/slab.h /* kmalloc() */ #include linux/fs.h/* everything... */ #include linux/errno.h /* error codes */ #include linux/types.h /* size_t */ #include linux/proc_fs.h #include linux/fcntl.h /* O_ACCMODE */ #include linux/seq_file.h #include linux/cdev.h #include linux/ioport.h
#include asm/system.h /* cli(), *_flags */ #include asm/uaccess.h /* copy_*_user */ #include asm/io.h
#define LED_NUM 4
struct led_dev { struct cdev dev void __iomem *base unsigned long offset }
struct led_dev led[4] dev_t dev = 0 int led_open(struct inode *inode, struct file *filp) { struct led_dev *led/* device information */
led = container_of(inode-i_cdev, struct led_dev, dev) filp-private_data = led/* for other methods */
return 0 /* success */ }
int led_release(struct inode *inode, struct file *filp) { return 0 }
ssize_t led_read(struct file *filp, char __user *buf, size_t count, loff_t *f_pos) { return 0 }
ssize_t led_write(struct file *filp, const char __user *buf, size_t count, loff_t *f_pos) { char data struct led_dev *led u32 value printk(KERN_INFO "debug by baikal: led dev write\n")
led = (struct led_dev *)filp-private_data copy_from_user(data,buf,count) if(data == '0') { printk(KERN_INFO "debug by baikal: led off\n") value = ioread32(led-base) iowrite32( value | 1led-offset, led-base) } else { printk(KERN_INFO "debug by baikal: led on\n") value = ioread32(led-base) iowrite32( value ~(1led-offset), led-base) } }
struct file_operations led_fops = { .owner =THIS_MODULE, .read = led_read, .write =led_write, //.ioctl =led_ioctl, .open = led_open, .release = led_release, }
static int led_init(void) { int result, i result = alloc_chrdev_region(dev, 0, LED_NUM,"LED") if (result 0) { printk(KERN_WARNING "LED: can't get major %d\n", MAJOR(dev)) return result }
for(i = 0i LED_NUMi++) { cdev_init( led[i].dev, led_fops) request_mem_region(0x56000014,0x4,"led") led[i].base = ioremap(0x56000014,0x4) led[i].offset = i + 5 //leds GPB5\6\7\8 led[i].dev.owner = THIS_MODULE led[i].dev.ops = led_fops result = cdev_add(led[i].dev,MKDEV(MAJOR(dev),i),1) if(result 0) { printk(KERN_ERR "LED: can't add led%d\n",i) return result } }
return 0 }
static void led_exit(void) { int i release_mem_region(0x56000014,0x4) for( i = 0i LED_NUMi++) { iounmap(led[i].base)
cdev_del(led[i].dev) } unregister_chrdev_region(dev, LED_NUM)
} module_init(led_init) module_exit(led_exit)
MODULE_AUTHOR("Baikal") MODULE_LICENSE("GPL") MODULE_DESCRIPTION("Simple LED Driver") lscpu命令,查看的是cpu的统计信息.\x0d\x0ablue@blue-pc:~$ lscpu\x0d\x0aArchitecture: i686#cpu架构\x0d\x0aCPU op-mode(s):32-bit, 64-bit\x0d\x0aByte Order:Little Endian #小尾序\x0d\x0aCPU(s):4 #总共有4核\x0d\x0aOn-line CPU(s) list: 0-3\x0d\x0aThread(s) per core:1 #每个cpu核,只能支持一个线程,即不支持超线程\x0d\x0aCore(s) per socket:4 #每个cpu,有4个核\x0d\x0aSocket(s): 1 #总共有1一个cpu\x0d\x0aVendor ID: GenuineIntel#cpu产商 intel\x0d\x0aCPU family:6\x0d\x0aModel: 42\x0d\x0aStepping: 7\x0d\x0aCPU MHz: 1600.000\x0d\x0aBogoMIPS: 5986.12\x0d\x0aVirtualization:VT-x#支持cpu虚拟化技术\x0d\x0aL1d cache: 32K\x0d\x0aL1i cache: 32K\x0d\x0aL2 cache: 256K\x0d\x0aL3 cache: 6144K\x0d\x0a \x0d\x0a 查看/proc/cpuinfo,可以知道每个cpu信息,如每个CPU的型号,主频等。\x0d\x0a#cat /proc/cpuinfo\x0d\x0aprocessor: 0\x0d\x0avendor_id: GenuineIntel\x0d\x0acpu family: 6\x0d\x0amodel: 42\x0d\x0amodel name: Intel(R) Core(TM) i5-2320 CPU @ 3.00GHz\x0d\x0a.....\x0d\x0a 上面输出的是第一个cpu部分信息,还有3个cpu信息省略了。\x0d\x0a \x0d\x0a 内存\x0d\x0a 概要查看内存情况\x0d\x0a free -m\x0d\x0a total used free sharedbuffers cached\x0d\x0a Mem: 3926 3651274 0 12404\x0d\x0a -/+ buffers/cache: 3235691\x0d\x0a Swap: 9536 31 9505\x0d\x0a 这里的单位是MB,总共的内存是3926MB。\x0d\x0a \x0d\x0a 查看内存详细使用\x0d\x0a# cat /proc/meminfo\x0d\x0aMemTotal:4020868 kB\x0d\x0aMemFree: 230884 kB\x0d\x0aBuffers:7600 kB\x0d\x0aCached: 454772 kB\x0d\x0aSwapCached: 836 kB\x0d\x0a.....\x0d\x0a \x0d\x0a 查看内存硬件信息\x0d\x0admidecode -t memory\x0d\x0a# dmidecode 2.11\x0d\x0aSMBIOS 2.7 present.\x0d\x0aHandle 0x0008, DMI type 16, 23 bytes\x0d\x0aPhysical Memory Array\x0d\x0aLocation: System Board Or Motherboard\x0d\x0a....\x0d\x0aMaximum Capacity: 32 GB\x0d\x0a....\x0d\x0aHandle 0x000A, DMI type 17, 34 bytes\x0d\x0a....\x0d\x0aMemory Device\x0d\x0aArray Handle: 0x0008\x0d\x0aError Information Handle: Not Provided\x0d\x0aTotal Width: 64 bits\x0d\x0aData Width: 64 bits\x0d\x0aSize: 4096 MB\x0d\x0a.....\x0d\x0a 我的主板有4个槽位,只用了一个槽位,上面插了一条4096MB的内存。\x0d\x0a \x0d\x0a 磁盘\x0d\x0a 查看硬盘和分区分布\x0d\x0a# lsblk\x0d\x0aNAME MAJ:MIN RM SIZE RO TYPE MOUNTPOINT\x0d\x0asda 8:00 465.8G 0 disk\x0d\x0a├—sda1 8:10 1G 0 part /boot\x0d\x0a├—sda2 8:20 9.3G 0 part [SWAP]\x0d\x0a├—sda3 8:30 74.5G 0 part /\x0d\x0a├—sda4 8:40 1K 0 part\x0d\x0a├—sda5 8:50 111.8G 0 part /home\x0d\x0a└—sda6 8:60 269.2G 0 part\x0d\x0a 显示很直观\x0d\x0a \x0d\x0a 如果要看硬盘和分区的详细信息\x0d\x0a# fdisk -l\x0d\x0aDisk /dev/sda: 500.1 GB, 500107862016 bytes\x0d\x0a255 heads, 63 sectors/track, 60801 cylinders, total 976773168 sectors\x0d\x0aUnits = sectors of 1 * 512 = 512 bytes\x0d\x0aSector size (logical/physical): 512 bytes / 4096 bytes\x0d\x0aI/O size (minimum/optimal): 4096 bytes / 4096 bytes\x0d\x0aDisk identifier: 0x00023728\x0d\x0a Device Boot Start End Blocks Id System\x0d\x0a/dev/sda1 *2048 2148351 1073152 83 Linux\x0d\x0a/dev/sda2 214835221680127 9765888 82 Linux swap / Solaris\x0d\x0a/dev/sda321680128 17793023978125056 83 Linux\x0d\x0a/dev/sda4 177932286 976771071 3994193935 Extended/dev/sda5 177932288 412305407 117186560 83 Linux\x0d\x0a/dev/sda6 412307456 976771071 282231808 83 Linux\x0d\x0a \x0d\x0a 网卡\x0d\x0a 查看网卡硬件信息\x0d\x0a# lspci | grep -i 'eth'\x0d\x0a02:00.0 Ethernet controller: Realtek Semiconductor Co., Ltd. RTL8111/8168B PCI Express Gigabit Ethernet controller (rev 06)\x0d\x0a \x0d\x0a 查看系统的所有网络接口\x0d\x0a# ifconfig -a\x0d\x0aeth0 Link encap:以太网 硬件地址 b8:97:5a:17:b3:8f \x0d\x0a .....\x0d\x0aloLink encap:本地环回 \x0d\x0a .....\x0d\x0a 或者是\x0d\x0aip link show\x0d\x0a1: lo: mtu 16436 qdisc noqueue state DOWN\x0d\x0alink/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00\x0d\x0a2: eth0: mtu 1500 qdisc pfifo_fast state UP qlen 1000\x0d\x0alink/ether b8:97:5a:17:b3:8f brd ff:ff:ff:ff:ff:ff\x0d\x0a \x0d\x0a 如果要查看某个网络接口的详细信息,例如eth0的详细参数和指标\x0d\x0a# ethtool eth0\x0d\x0aSettings for eth0:\x0d\x0aSupported ports: [ TP MII ]\x0d\x0aSupported link modes: 10baseT/Half 10baseT/Full\x0d\x0a100baseT/Half 100baseT/Full\x0d\x0a1000baseT/Half 1000baseT/Full #支持千兆半双工,全双工模式\x0d\x0aSupported pause frame use: No\x0d\x0aSupports auto-negotiation: Yes #支持自适应模式,一般都支持\x0d\x0aAdvertised link modes: 10baseT/Half 10baseT/Full\x0d\x0a100baseT/Half 100baseT/Full\x0d\x0a1000baseT/Half 1000baseT/Full\x0d\x0aAdvertised pause frame use: Symmetric Receive-only\x0d\x0aAdvertised auto-negotiation: Yes #默认使用自适应模式\x0d\x0aLink partner advertised link modes: 10baseT/Half 10baseT/Full\x0d\x0a 100baseT/Half 100baseT/Full\x0d\x0a.....\x0d\x0aSpeed: 100Mb/s #现在网卡的速度是100Mb,网卡使用自适应模式,所以推测路由是100Mb,导致网卡从支 持千兆,变成要支持百兆\x0d\x0aDuplex: Full #全双工\x0d\x0a.....\x0d\x0aLink detected: yes#表示有网线连接,和路由是通的\x0d\x0a\x0d\x0a其他\x0d\x0a 查看pci信息,即主板所有硬件槽信息。\x0d\x0alspci\x0d\x0a00:00.0 Host bridge: Intel Corporation 2nd Generation Core Processor Family DRAM Controller (rev 09) #主板芯片\x0d\x0a00:02.0 VGA compatible controller: Intel Corporation 2nd Generation Core Processor Family Integrated Graphics Controller (rev 09) #显卡\x0d\x0a00:14.0 USB controller: Intel Corporation Panther Point USB xHCI Host Controller (rev 04) #usb控制器\x0d\x0a00:16.0 Communication controller: Intel Corporation Panther Point MEI Controller #1 (rev 04)\x0d\x0a00:1a.0 USB controller: Intel Corporation Panther Point USB Enhanced Host Controller #2 (rev 04)\x0d\x0a00:1b.0 Audio device: Intel Corporation Panther Point High Definition Audio Controller (rev 04) #声卡\x0d\x0a00:1c.0 PCI bridge: Intel Corporation Panther Point PCI Express Root Port 1 (rev c4) #pci 插槽\x0d\x0a00:1c.2 PCI bridge: Intel Corporation Panther Point PCI Express Root Port 3 (rev c4)\x0d\x0a00:1c.3 PCI bridge: Intel Corporation Panther Point PCI Express Root Port 4 (rev c4)\x0d\x0a00:1d.0 USB controller: Intel Corporation Panther Point USB Enhanced Host Controller #1 (rev 04)\x0d\x0a00:1f.0 ISA bridge: Intel Corporation Panther Point LPC Controller (rev 04)\x0d\x0a00:1f.2 IDE interface: Intel Corporation Panther Point 4 port SATA Controller [IDE mode] (rev 04) #硬盘接口\x0d\x0a00:1f.3 SMBus: Intel Corporation Panther Point SMBus Controller (rev 04)\x0d\x0a00:1f.5 IDE interface: Intel Corporation Panther Point 2 port SATA Controller [IDE mode] (rev 04) #硬盘接口\x0d\x0a02:00.0 Ethernet controller: Realtek Semiconductor Co., Ltd. RTL8111/8168B PCI Express Gigabit Ethernet controller (rev 06) #网卡\x0d\x0a03:00.0 PCI bridge: Integrated Technology Express, Inc. Device 8893 (rev 41)\x0d\x0a 如果要更详细的信息:lspci -v 或者 lspci -vv\x0d\x0a 如果要看设备树:lscpi -t\x0d\x0a \x0d\x0a 查看bios信息\x0d\x0a# dmidecode -t bios\x0d\x0a......\x0d\x0aBIOS Information\x0d\x0aVendor: American Megatrends Inc.\x0d\x0aVersion: 4.6.5\x0d\x0aRelease Date: 04/25/2012\x0d\x0a.......\x0d\x0aBIOS Revision: 4.6\x0d\x0a......\x0d\x0a dmidecode以一种可读的方式dump出机器的DMI(Desktop Management Interface)信息。这些信息包括了硬件以及BIOS,既可以得到当前的配置,也可以得到系统支持的最大配置,比如说支持的最大内存数等。\x0d\x0a 如果要查看所有有用信息\x0d\x0a dmidecode -q\x0d\x0a 以上是linux查看硬件信息的所有命令,可以查看CPU、硬盘、网卡、磁盘等硬件的信息。欢迎分享,转载请注明来源:内存溢出 原文地址:https://outofmemory.cn/yw/7143433.html |
CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3 |